Vuetify 自定义 SVG 图标

Vite

Vuetify

vite-plugin-svg-icons

  1. 首先配置 svg 插件https://github.com/vbenjs/vite-plugin-svg-icons/blob/main/README.zh_CN.md

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    // vite.config.js
    import { defineConfig, loadEnv } from "vite";
    import legacy from "@vitejs/plugin-legacy";
    import { createVuePlugin } from "vite-plugin-vue2";
    import viteCompression from "vite-plugin-compression";
    import path from "path";
    import WindiCSS from "vite-plugin-windicss";
    import { createSvgIconsPlugin } from "vite-plugin-svg-icons";
    import { VuetifyResolver } from "unplugin-vue-components/resolvers";
    import Components from "unplugin-vue-components/vite";
    const HOST = "0.0.0.0";

    export default ({ mode }) => {
    return defineConfig({
    base: "./",
    server: {
    host: HOST,
    port: loadEnv(mode, process.cwd()).VITE_APP_PORT,
    },
    resolve: {
    alias: {
    "@": path.resolve(__dirname, "/src"),
    },
    },
    plugins: [
    createVuePlugin(/* options */),
    legacy({
    targets: ["ie >= 11"],
    additionalLegacyPolyfills: ["regenerator-runtime/runtime"],
    }),
    WindiCSS(),
    viteCompression(),
    // 配置插件
    createSvgIconsPlugin({
    iconDirs: [path.resolve(process.cwd(), "src/assets/icons")],
    symbolId: "icon-[dir]-[name]",
    }),
    Components({
    resolvers: [VuetifyResolver()],
    }),
    ],
    });
    };
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    // SvgIcon.vue
    <template>
    <svg aria-hidden="true">
    <use :xlink:href="symbolId" :fill="color" />
    </svg>
    </template>

    <script>
    export default {
    name: "SvgIcon",
    props: {
    prefix: {
    type: String,
    default: "icon",
    },
    name: {
    type: String,
    required: true,
    },
    color: {
    type: String,
    default: "#333",
    },
    },
    computed: {
    symbolId() {
    return `#${this.prefix}-${this.name}`;
    },
    },
    };
    </script>
  2. 配置 vuetify

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    import Vue from "vue";
    import Vuetify from "vuetify/lib/framework";
    import "material-design-icons-iconfont/dist/material-design-icons.css";

    import SvgIcon from "@/components/SvgIcon.vue";
    import ids from "virtual:svg-icons-names"; // 引入全部图标名

    Vue.use(Vuetify);
    export default new Vuetify({
    icons: {
    iconfont: "md", // 'mdi' || 'mdiSvg' || 'md' || 'fa' || 'fa4' || 'faSvg'
    values: (() => {
    const icons = {}; //处理图标,配置到vuetify
    ids.forEach(id => {
    icons[id.substr(5)] = {
    component: SvgIcon,
    props: {
    name: id.substr(5),
    },
    };
    });
    return icons;
    })(),
    },
    });
  3. 使用图标

    1. 把 svg 放到 assets/icons 目录

      image-20220720164727101

    2. icon 中使用,在名称前边加上$

      1
      2
      3
      4
      5
      6
      <v-btn icon>
      <v-icon>$dir-薯条2</v-icon>
      </v-btn>
      <v-btn icon>
      <v-icon>$薯条</v-icon>
      </v-btn>
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      <v-text-field v-model="search" label="" placeholder="请输入名称" solo hide-details prepend-inner-icon="menu" clearable style="width: 390px;" append-outer-icon="$薯条">
      <template v-slot:append>
      <v-btn icon class="ml-2" small color="primary">
      <v-icon>$dir-薯条2</v-icon>
      </v-btn>
      </template>
      <template v-slot:prepend-inner>
      <v-btn icon small color="primary" @click="$bus.emit('targetDrawer')">
      <v-icon>menu</v-icon>
      </v-btn>
      </template>
      </v-text-field>

      image-20220720165136865